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To what extent should one trust a statement that a program is free of Trojan 
horses? Perhaps it is more important to trust the people who wrote the 
software. 



KEN THOMPSON 



INTRODUCTION 

I thank the ACM for this award. I can’t help but feel 
that I am receiving this honor for timing and serendip- 
ity as much as technical merit. UNIX 1 swept into popu- 
larity with an industry-wide change from central main- 
frames to autonomous minis. I suspect that Daniel Bob- 
row [1] would be here instead of me if he could not 
afford a PDP-10 and had had to “settle” for a PDP-11. 
Moreover, the current state of UNIX is the result of the 
labors of a large number of people. 

There is an old adage. “Dance with the one that 
brought you,” which means that I should talk about 
UNIX. I have not worked on mainstream UNIX in many 
years, yet I continue to get undeserved credit for the 
work of others. Therefore, I am not going to talk about 
UNIX, but I want to thank everyone who has contrib- 
uted. 

That brings me to Dennis Ritchie. Our collaboration 
has been a thing of beauty. In the ten years that we 
have worked together, I can recall only one case of 
miscoordination of work. On that occasion, I discovered 
that we both had written the same 20-line assembly 
language program. I compared the sources and was as- 
tounded to find that they matched character-for-char- 
acter. The result of our work together has been far 
greater than the work that we each contributed. 

I am a programmer. On my 1040 form, that is what I 
put down as my occupation. As a programmer, I write 


1 UNIX is a trademark of AT&T Bell Laboratories. 
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programs. I would like to present to you the cutest 
program I ever wrote. I will do this in three stages and 
try to bring it together at the end. 

STAGE I 

In college, before video games, we would amuse our- 
selves by posing programming exercises. One of the 
favorites was to write the shortest self-reproducing pro- 
gram. Since this is an exercise divorced from reality, 
the usual vehicle was FORTRAN. Actually, FORTRAN 
was the language of choice for the same reason that 
three-legged races are popular. 

More precisely stated, the problem is to write a 
source program that, when compiled and executed, will 
produce as output an exact copy of its source. If you 
have never done this, I urge you to try it on your own. 
The discovery of how to do it is a revelation that far 
surpasses any benefit obtained by being told how to do 
it. The part about “shortest” was just an incentive to 
demonstrate skill and determine a winner. 

Figure 1 shows a self-reproducing program in the C 3 
programming language. (The purist will note that the 
program is not precisely a self-reproducing program, 
but will produce a self-reproducing program.) This en- 
try is much too large to win a prize, but it demonstrates 
the technique and has two important properties that I 
need to complete my story: 1) This program can be 
easily written by another program. 2) This program can 
contain an arbitrary amount of excess baggage that will 
be reproduced along with the main algorithm. In the 
example, even the comment is reproduced. 
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char s[ ] = | 

'V', 

'O', 

'\n', 

r. 

'V?', 

'Vj', 

r. 

'V)', 

(213 lines deleted) 
0 


/* 

• The string s is a 

* representation of the body 

• of this program from '0' 

« to the end. 

*/ 

main( ) 

I 

int /; 

printf(“char\ts[ ]=|Vt"); 
for(/=0; s[/]; /++) 

printf(“\r%d, X/j", s[/]); 
printff%s", s); 

I 

Here are some simple transliterations to allow 
a non-C programmer to read this code. 

= assignment 

== equal to .EQ. 

!= not equal to .NE. 

++ increment 

'x ' single character constant 
“xxx” multiple character string 

%d format to convert to decimal 

%s format to convert to string 
V tab character 

\n newline character 


FIGURE 1. 


STAGE II 

The C compiler is written in C. What I am about to 
describe is one of many “chicken and egg” problems 
that arise when compilers are written in their own lan- 
guage. In this case, I will use a specific example from 
the C compiler. 

C allows a string construct to specify an initialized 
character array. The individual characters in the string 
can be escaped to represent unprintable characters. For 
example, 

“Hello worldXn” 

represents a string with the character “\n," representing 
the new line character. 

Figure 2.1 is an idealization of the code in the C 
compiler that interprets the character escape sequence. 
This is an amazing piece of code. It “knows” in a com- 
pletely portable way what character code is compiled 
for a new line in any character set. The act of knowing 


then allows it to recompile itself, thus perpetuating the 
knowledge. 

Suppose we wish to alter the C compiler to include 
the sequence "\v" to represent the vertical tab charac- 
ter. The extension to Figure 2.1 is obvious and is pre- 
sented in Figure 2.2. We then recompile the C com- 
piler, but we get a diagnostic. Obviously, since the bi- 
nary version of the compiler does not know about "\v," 
the source is not legal C. We must “train” the compiler. 
After it “knows” what “ \v ” means, then our new 
change will become legal C. We look up on an ASCII 
chart that a vertical tab is decimal 11. We alter our 
source to look like Figure 2.3. Now the old compiler 
accepts the new source. We install the resulting binary 
as the new official C compiler and now we can write 
the portable version the way we had it in Figure 2.2. 

This is a deep concept. It is as close to a “learning” 
program as I have seen. You simply tell it once, then 
you can use this self-referencing definition. 

STAGE III 

Again, in the C compiler, Figure 3.1 represents the high 
level control of the C compiler where the routine “com- 


c = next( ); 
if(c != '\Y) 

retum(c); 
c = next( ); 
if(c == AX') 

retum('W'); 
if(c - 'n ') 

retum('Xn'); 


FIGURE 2.2. 


c = next( ); 
if(c != '\Y) 

retum(c); 
c = next( ); 
if(c == AX') 

return('W'); 
if(c == ’n') 

retum('\/7'); 
if(c == V') 

retum('\v'); 


FIGURE 2.1. 


c = next( ); 
if(c != AY) 

return(c); 
c = next( ); 
if (c == AX') 

return(AX'); 
if(c == ' n ') 

retum(An'); 
if(c == 'v') 

return(ll); 


FIGURE 2.3. 
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pile” is called to compile the next line of source. Figure 
3.2 shows a simple modification to the compiler that 
will deliberately miscompile source whenever a partic- 
ular pattern is matched. If this were not deliberate, it 
would be called a compiler “bug.” Since it is deliberate, 
it should be called a “Trojan horse.” 

I The actual bug I planted in the compiler would 

match code in the UNIX “login" command. The re- 
placement code would miscompile the login command 
so that it would accept either the intended encrypted 
password or a particular known password. Thus if this 
code were installed in binary and the binary were used 

I to compile the login command, I could log into that 
system as any user. 

Such blatant code would not go undetected for long. 
Ty.m the most casual perusal of the source of the C 
compiler would raise suspicions. 

The final step is represented in Figure 3.3. This sim- 

I ply adds a second Trojan horse to the one that already 
exists. The second pattern is aimed at the C compiler. 
The replacement code is a Stage I self-reproducing pro- 
gram that inserts both Trojan horses into the compiler. 
This requires a learning phase as in the Stage II exam- 
ple. First we compile the modified source with the nor- 
mal C compiler to produce a bugged binary. We install 
this binary as the official C. We can now remove the 
bugs from the source of the compiler and the new bi- 
nary will reinsert the bugs whenever it is compiled. Of 

[ course, the login command will remain bugged with no 
trace in source anywhere. 


compiles) 
char «s; 


FIGURE 3.1. 


compile(s) 
char *s; 


if(match(s, “pattern”)) I 
compilefbug”); 
return; 


FIGURE 3.2. 


compiles) 
char «s; 


if(match(s, “patteml")) | 
compile (“bugl”); 
return; 

I 

if(match(s, “pattern 2")) | 
compile (“bug 2"); 
return; 


FIGURE 3.3. 
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MORAL 

The moral is obvious. You can’t trust code that you did 
not totally create yourself. (Especially code from com- 
panies that employ people like me.) No amount of 
source-level verification or scrutiny will protect you 
from using untrusted code. In demonstrating the possi- 
bility of this kind of attack, I picked on the C compiler. 

I could have picked on any program-handling program 
such as an assembler, a loader, or even hardware mi- 
crocode. As the level of program gets lower, these bugs 
will be harder and harder to detect. A well-installed 
microcode bug will be almost impossible to detect. 

After trying to convince you that I cannot be trusted, 

I wish to moralize. I would like to criticize the press in 
its handling of the “hackers,” the 414 gang, the Dalton 
gang, etc. The acts performed by these kids are vandal- 
ism at best and probably trespass and theft at worst. It 
is only the inadequacy of the criminal code that saves 
the hackers from very serious prosecution. The compa- 
nies that are vulnerable to this activity, (and most large 
companies are very vulnerable) are pressing hard to 
update the criminal code. Unauthorized access to com- 
puter systems is already a serious crime in a few states 
and is currently being addressed in many more state 
legislatures as well as Congress. 

There is an explosive situation brewing. On the one 
hand, the press, television, and movies make heros of 
vandals by calling them whiz kids. On the other hand, 
the acts performed by these kids will soon be punisha- 
ble by years in prison. 

I have watched kids testifying before Congress. It is 
clear that they are completely unaware of the serious- 
ness of their acts. There is obviously a cultural gap. The 
act of breaking into a computer system has to have the 
same social stigma as breaking into a neighbor’s house. 
It should not matter that the neighbor’s door is un- 
locked. The press must learn that misguided use of a 
computer is no more amazing than drunk driving of an 
automobile. 


Acknowledgment. I first read of the possibility of such 
a Trojan horse in an Air Force critique [4] of the secu- 
rity of an early implementation of Multics. I cannot find 
a more specific reference to this document. I would 
appreciate it if anyone who can supply this reference 
would let me know. 
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Principles in 1973 and a revised ver- 
sion subsequently appeared in the 
July 1974 issue of Communications. 
This paper received the ACM award 
for best paper in programming lan- 
guages and systems in 1974. 

According to the Turing Award 
selection committee, “The success 
of the UNIX system stems from its 
tasteful selection of a few key ideas 
and their elegant implementation. 
The model of the UNIX system has 
led a generation of software design- 
ers to new ways of thinking about 
programming. The genius of the 
UNIX system is its framework, 
which enables programmers to 
stand on the work of others.” 

The award is the Association’s 
highest recognition of technical 
contributions to the computing 
community. It honors Alan M. 
Turing, the English mathematician 
who made major contributions to 
the computing sciences. 

Ritchie and Thompson gave sepa- 
rate lectures at the conference. 
Ritchie focused on the nature of the 
environment at Bell Labs that made 
development of UNIX possible. 
Thompson reflected on the question 
of how much one can trust software 
in contradistinction to people who 
write it. 


Ken L. Thompson 


Dennis Ritchie 


The ACM A.M. Turing Award for 
1983 was presented to Dennis M. 
Ritchie and Ken L. Thompson of 
AT&T Bell Laboratories at the Asso- 
ciation’s Annual Conference in 
October for their development and 
implementation of the UNIX oper- 
ating system. 

The UNIX time-sharing system 
was conceived by Thompson and 
developed jointly with Ritchie in the 
late 1960s. A key contribution to the 
portability of the UNIX system was 
the development by Ritchie of the 
C Programming Language. 

Their seminal paper, “The UNIX 
Time-Sharing System,” was orig- 
inally presented at the Fourth ACM 
Symposium on Operating Systems 
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